# Digital Electronics

2<sup>nd</sup> workshop session

### Verilog constructs

- The Verilog files have the extension .v
- Verilog is case sensitive
- Verilog line of code terminates with semicolon ';'
- Comments like in C by "//" for single line and "/\*....\*/" for multiple lines
- The code here executes in parallel unlike the normal programing languages

### Numeric constants

• For a single wire, the data value that can appear on it is the following:

| Representation   | Value |
|------------------|-------|
| "Low" logic zero | 0     |
| "High" logic one | 1     |
| High impedance   | Z or? |
| unknown          | X     |

- 'X' is used by the simulators only to represent the wires that are not initialized by a known value
- By default, constants are not signed.
- For a couple of wires "BUS", we can assign data to the whole bus without assigning wire by wire:

| Syntax                   | Symbol | Numbering system  |
|--------------------------|--------|-------------------|
| Bus = 100 or Bus = 'd100 | d      | Decimal (default) |
| Bus = 'b0110_0100        | b      | Binary            |
| Bus = '0144              | О      | Octal             |
| BUS = 'h64               | h      | Hexadecimal       |

### Numeric constants

• General numbering representation

### [sign][size]'[base][value]

| Number    | Stored value | Comment                 |
|-----------|--------------|-------------------------|
| 5'b11010  | 11010        |                         |
| 5'b11_010 | 11010        | _ignored                |
| 5′032     | 11010        |                         |
| 5'hla     | 11010        |                         |
| 5'd26     | 11010        |                         |
| 5'b0      | 00000        | 0 extended              |
| 5'bl      | 00001        | 0 extended              |
| 5'bz      | ZZZZZ        | z extended              |
| 5'bx      | xxxxx        | x extended              |
| 5'bx01    | xxx01        | x extended              |
| -5′Ъ00001 | 11111        | 2's complement of 00001 |

SP21 - ECE 3300 - Digital Circuit Design Using Verilog

### • Bitwise operators

- Performs bit by bit evaluation between two operands
- 2'b01 & 2'b11 = {0&1,1&1}
- $\sim$  (3'b011) = 3'b100

### • <u>Logical operators</u>

- Returns one-bit 1 or 0 (true or false)
- -- !(2'b01) = 0
- "==" tests logical equality for '1' and '0' only all other will result in 'X'
- "===" tests 4-state logical equality (1,0,Z,X)

#### bitwise

| NOT  | ~a     |
|------|--------|
| AND  | A & b  |
| OR   | A   B  |
| XOR  | A^B    |
| XNOR | A ~^ B |
|      | A ^~ B |

#### Logical

| NOT                                               | !A                 |
|---------------------------------------------------|--------------------|
| AND                                               | A && b             |
| OR                                                | A     B            |
| If X or Z in bits returns X                       | A == B<br>A != B   |
| Return 0 or 1 only based on bit-by-bit comparison | A === B<br>A !== B |

### • reduction operators

- Evaluate bit-by-bit of a vector (unary operators)
- -- &(4'b0111) = {0&1&1&1} = 0
- Arithmetic operators
  - Used to perform arithmetic operations
- Shift operators
  - Shift the first operand by the number specified by the second operand

| Arithmetic shift <<< , >>> | Logical shift << , >> |
|----------------------------|-----------------------|
| 3'b110 >>> 1 = 3'b111      | 3'b110 >> 1 = 3'b011  |

#### reduction

| AND  | &A  |
|------|-----|
| NAND | ~&A |
| OR   | A   |
| NOR  | ~ A |
| XOR  | ^A  |

#### Arithmetic

| addition       | + |
|----------------|---|
| subtraction    | - |
| multiplication | * |
| division       | / |
| modulus        | % |

- Relational operators
  - Compares 2 operands and returns 1 or 0
- Concatenation and replication operators
  - {2'b00,2'b11,2'b00,2'b11} = 8'b00\_11\_00\_11
  - {{4{2'b11}}},8'b0} = 16'b1111\_1111\_0000\_0000
  - Concatenation and replication have very useful applications and they are very fast and e
- Conditional operator
  - Condition? True: False

#### relational

| Less than             | <  |
|-----------------------|----|
| Less than or equal    | <  |
|                       | =  |
| Greater than          | >  |
| Greater than or equal | >  |
|                       | Ш  |
| Equal to              | =  |
| asy to use            | Ш  |
| Not equal to          | != |

• Operators' priority in Verilog

| Verilog Operator | Name                      | Functional Group      |  |
|------------------|---------------------------|-----------------------|--|
| []               | bit-select or part-select |                       |  |
| ()               | parenthesis               |                       |  |
| !                | logical negation          | logical               |  |
| ~<br>&           | negation<br>reduction AND | bit-wise<br>reduction |  |
| - X              | reduction OR              | reduction             |  |
| ~&               | reduction NAND            | reduction             |  |
| ~                | reduction NOR             | reduction             |  |
| ^'               | reduction XOR             | reduction             |  |
| ~^ or ^~         | reduction XNOR            | reduction             |  |
| +                | unary (sign) plus         | arithmetic            |  |
| -                | unary (sign) minus        | arithmetic            |  |
| {}               | concatenation             | concatenation         |  |
| {{ }}            | replication               | replication           |  |
| *                | multiply                  | arithmetic            |  |
| /                | divide                    | arithmetic            |  |
| %                | modulus                   | arithmetic            |  |
| +                | binary plus               | arithmetic            |  |
| -                | binary minus              | arithmetic            |  |
| <<               | shift left                | shift                 |  |
| >>               | shift right               | shift                 |  |
| >                | greater than              | relational            |  |
| >=               | greater than or equal to  |                       |  |
| <                | less than                 | relational            |  |
| <=               | less than or equal to     | relational            |  |
| ==               | logical equality          | equality              |  |
| !=               | logical inequality        | equality              |  |
| ===              | case equality             | equality              |  |
| !==              | case inequality           | equality              |  |
| &                | bit-wise AND              | bit-wise              |  |
| ^                | bit-wise XOR              | bit-wise              |  |
| ^~ or ~^         | bit-wise XNOR             | bit-wise              |  |
|                  | bit-wise OR               | bit-wise              |  |
| &&               | logical AND               | logical               |  |
|                  | 1 · 1 OD                  | 1 . 1                 |  |
|                  | logical OR                | logical               |  |

```
module AND(A,B,out);
    input A,B;
    output out;
    assign out = A&B;
endmodule
```

```
module MUX2x1(A,B,,sel,out);
    input A,B,sel;
    output out;
    assign out = sel?B:A;
endmodule
```

### • <u>Module</u>

- Basic building block in Verilog
- Each module has directional ports (input, output, inout) used to communicate with the module
- A module can be an element or a collection of lower-level design blocks
- A module declared with keyword module
- A corresponding keyword endmodule must appear at the end of the module
- Ports are declared as wires by default

#### • wire

- Wire represents a physical wire that is used inside a circuit or module
- Wire doesn't store its value and so must be continuously driven
- Bus is an array of wires moving together in parallel to transfer data
- Note that we can type the width of the bus in the square brackets like this ex. [0:3] but here the MSB will be [0] and LSB will be [3] which is little bit confusing, so we used to write it in this format [MSB:LSB]

```
wire a,b,c;  //three 1-bit wires
wire [31:0] data;  //a 32-bit bus
wire [7:0] b1,b2,b3,b4; //four 8-bit busses
wire [W-1:0] in;  //parameterized bus
```

- Continuous assignment
  - Continuous assignment are used to describe a behavior equivalent to combinational logic
  - Keyword assign is used for the assignment of wires
  - Continuous assignments are continuously evaluated so when ever the RHS changes, the LFS is updated (typical combinational logic)

- Module instantiation
  - Module instances provide the ability to reuse any given module inside other modules
  - Remember in design flow we implemented 4x1Mux using three 2x1Muxs
  - There are two ways for port mapping (by name, by order)

```
module MUX4X1(A,B,C,D,Sel,Out);

input A,B,C,D;
input [1:0] Sel;
output Out;

wire m0,m1;

MUX2X1 M0 (.A(A),.B(B),.Sel(Sel[0]),.Out(m0));
mUX2X1 M1 (.A(C),.B(D),.Sel(Sel[0]),.Out(m1));
MUX2X1 M3 (.A(m0),.B(m1),.Sel(Sel[1]),.Out(Out));

endmodule
port
connection list
```

```
module MUX2X1(A,B,Sel,Out);

input A,B,Sel;
output Out;

assign Out = Sel?B:A;

endmodule
```

Module name

Instance name

### Coding styles

### 1. Gate-level modeling

- The module is implemented using the built-in logic gates and the interconnection between them
- This level can be used only with small circuits

### 2. Dataflow modeling

— Designer must know how the data flow within the design

### 3. <u>Structural modeling</u>

- The same as gate level modeling but doesn't use built-in logic gates
- It instantiate predefined modules by the designer and connect them together to build a larger circuit

### 4. <u>Behavioral modeling</u>

- This is the most important modeling that designers use, it can model more complex designs and sequential blocks must be modeled with this one
- Designer needs only to know the functionality of the circuit and describe its behavior in a C-like code
- It will be studied in detail in session 2.

### Coding styles

#### Gate level

```
module Mux2x1(A,B,sel,out);
    input A,B,sel;
    output out;
    wire w1,w2,sel_bar;
    not(sel_bar,sel);
    and(w1,A,sel);
    and(w2,B,sel_bar);
    or(w1,w2);
endmodule
```

#### structural

```
module Mux4x1(in1,in2,in3,in4,sel,out);
    input in1,in2,in3,in4;
    input [1:0] sel;
    output reg out;

    wire w1,w2;

MUX2x1 m0 (.A(in1),.B(in2),.sel(sel[0]),.out(w1));
    MUX2x1 m1 (.A(in3),.B(in4),.sel(sel[0]),.out(w2));
    MUX2x1 m2 (.A(w1),.B(w2),.sel(sel[1]),.out(out));
endmodule
```

#### Data flow

```
module Mux2x1(A,B,sel,out);
    input A,B,sel;
    output out;
    assign out = sel?A:B;
endmodule
```

#### behavioral

```
module Mux2x1(A,B,sel,out);

input A,B,sel;
output reg out;

always@(*)begin
    case(sel)
    1'b0: out = B;
    1'b1: out = A;
    default out = 1'b1;
    endcase
end
```

## Lab part

Designing full adder and four-bit adder

### Requirements

- Design full adder using gate-level modeling
- Write the testbench
- Simulate it on modalism
- Synthesis the circuit and then burn in on FPGA
- · Repeat for 4-bit adder using full adders that you designed before
- Challenge: try to design generic n-bit adder using full adders, hint: you need to search about *generate* block and you will use *for* loops and keyword *parameter*

### Agenda

- Quick Revision
- > Introduction
- Verilog abstraction levels
- > combinational circuits
- > Procedural blocks
  - always block
  - Initial block(test benches)
- > Procedural statements
- > Incomplete assignment
- > Lab

### Quick REVISION

- □ Coding styles.
- □ module instantiation.
- ☐ Continuous assignment(assign statement).
- ☐ Operators.

```
1 module MUX2X1(A,B,Sel,Out);
2
3    input A,B,Sel;
4    output Out;
5
6    assign Out = Sel?B:A;
7
8 endmodule
```

### Introduction

- Verilog is a HARDWARE DESCRIPTION LANGUAGE (HDL). It is a language used for describing a digital system like a network switch or a microprocessor or a memory or a flip—flop. It means, by using an HDL we can describe any digital hardware at any level. Designs, which are described in HDL are independent of technology, very easy for designing and debugging, and are normally more useful than schematics, particularly for large circuits.
- ☐ When we write HDL code, we must think about our final HW and what will the output of my code ??
- Always ask your self what my code will synthesis for, and it will verify my specs or not??
- why we need to ask ourselves this questions in this session we will know.



### Modeling Combinational Logic

### ☐ Combinational logic

The **output** at any instant of time is determined directly from the **present combination** of inputs.





- ☐ Examples
  - Adder, Subtractor, Decoder, Encode, Multiplexer, Demultiplexer, etc.
  - > Outputs in combinational Circuits are a combination of gates all must be the same logic family

## Behavioural Level Verilog

Procedural blocks

### Behavioural level

- ☐ It is a model that describe the design algorithm in high-level of abstraction with little concern for actual hardware implementation.
- □ Easier to write and understand.
- □ Not always synthesizable.

Note: we design our circuits using RTL (register transfer logic) which is in between behavioural level and Gate level simulation(GLS).

### Signals Types

### wire

- ☐ It describes a simple physical connection between two structural elements.
- Any assignments using assign statement must assign to a wire variable.
- ☐ All inputs and output ports are by default defined as a wire type.
- ☐ It may take be on of this 4 values : -0 (Logic 0) -1(Logic 1)
   X(unknown) Z(high impedance)

For example: wire [15:0] data; // declaration of 16-bit signal of type wire.

### Signals Types

### reg

- ☐ It is a variable which may or mayn't describe a hardware register.
- ☐ Any assignments in an always block must assign to a reg variable.
- ☐ Used for applying stimulus in testbench.
- ☐ It may take be on of this 4 values : -0 (Logic 0) -1(Logic 1)
   X(unknown) Z(high impedance)

For example: reg [15:0] data; // declaration of 16-bit signal of type reg.

### Procedural blocks

### always block

□ syntax

```
always @(sensitivity list)

begin

statements;
end
```

hint: begin and end are only needed one statement inside always block.

note: all variables assigned inside always block are of type reg.

- ➤ In always block the statements are executed in sequence(procedurally).
- The code inside always block is executed when one of the sensitivity list changes.
- > all always blocks are executed in parallel.
- > Used to describe both combinational and sequential logic.

### Sensitivity List

- ☐ The [sensitivity list] term is a list of signals and events to which the always block responds (i.e., is "sensitive to").
- ☐ For a combinational circuit, all the input signals should be included in this list.



```
module example(
  input A,B,
  output reg out_and, out_or
);

always@(A or B)
  out_and = A&B;

always@(A or B)
  out_or = A|B;

endmodule
```

### Design Example

☐ Half adder circuit



```
module halfadder(
input A,B,
 output reg sum, carry
);
always@(A or B) begin
 sum = A^B;
 carry = A&B;
end
endmodule
```

### IF Statement

□ Syntax

```
if (boolean expression)
begin
  procedural statemen;
end
else if (boolean expression)
 procedural statemen;
else
begin
  procedural statemen;
end
```

➤ If else statement are synthesized by generating a multiplexer for any reg assigned within if statements the selection of each MUX is driven by logic determined by the if condition (priority).



### Design Example

☐ Design a simple 2×4 decoder as shown in the below truth table.

| a[1] | a[0] | Y    |
|------|------|------|
| 0    | 0    | 0001 |
| 0    | 1    | 0010 |
| 1    | 0    | 0100 |
| 1    | 1    | 1000 |

Note that: In Verilog-2001, a special notation, @(\*) is introduced to implicitly include all the relevant input signals.

```
module decoder(
 input [1:0] a,
 output reg [3:0] Y
always@(*) begin
 if(a == 2'b00)
  Y = 4'b0001;
 else if(a == 2'b01)
  Y = 4'b0010;
 else if(a == 2'b10)
  Y = 4'b0100;
 else
  Y = 4'b1000;
end
endmodule
```

### Case Statement

☐ Syntax

```
case (expression)
 value1: begin
          statement;
          end
 value2 : begin
          statement;
          end
 default : begin
          statement;
          end
endcase
```

- A case statement is a multiway decision statement that compares the expression with a number of [value] expressions.
- ☐ The execution jumps to the branch whose [value] matches the current value of the expression.
- Hint: case statement is used for mutually exclusive cases and only inside procedural blocks.



### Design Example

☐ Design a simple 2×4 decoder as shown in the below truth table.

| a[1] | a[0] | Y    |
|------|------|------|
| 0    | 0    | 0001 |
| 0    | 1    | 0010 |
| 1    | 0    | 0100 |
| 1    | 1    | 1000 |

```
module decoder(
input [1:0] a,
output reg [3:0] Y
always@(*) begin
case(a)
  2'b00 : Y = 4'b0001 ;
  2'b01 : Y = 4'b0010;
  2'b10 : Y = 4'b0100 ;
  2'b11 : Y = 4'b1000;
  default : Y = 4'b0000;
endcase
end
endmodule
```

# Incomplete assignment in Verilog For if statements and case statements

### If statement incomplete assignment

- □ What happens if we didn't assign a signal value in if statement?????
- ☐ Incomplete assignment happens when a signal is assigned under certain input condition and not assigned under the other input conditions.
- □ Example:

hint: incomplete assignment causes unintentional latch

```
if(X)
 begin
        A=1'b1;
 end
else if (Y)
 begin
        B=1'b1;
 end
else
 begin
        A=1'b0:
        B=1'b0;
 end
```

### Case statement incomplete assignment

- □ What happens if we didn't assign a signal value in one of cases in case statement????
- □ What if we didn't write default branch????
- ☐ Incomplete assignment happens when a signal is assigned under certain case branch and not assigned under the other case branches or when missing default branches.

hint: incomplete assignment causes unintentional latch.

### Case statement incomplete assignment

### • Examples:

```
case (SEL)
2'b00:begin
       A=1'b1;
       B=1'b=0;
      end
 2'bl0:begin
       A=1'b1;
       B=1'b=0;
      end
 2'bl1:begin
       A=1'b1;
       B=1'b=0;
      end
 endcase
```

```
case (SEL)
 2'b00:begin
       A=1'b1;
        B=1'b=0;
       end
2'bl0:begin
       A=1'b1:
        B=1'b=0;
       end
 2'bll:begin
       A=1'b1;
       end
default:begin
       A=1'b0;
        B=1'b=0;
       end
 endcase
```

## Guidelines For Designing

- ☐ Assign a variable only in a single always block.
- ☐ Use @\*to include all inputs automatically in the sensitivity list.
- ☐ Make sure that all branches of the if and case statements are included.
- ☐ Make sure that the outputs are assigned in all branches.
- ☐ One way to satisfy the two previous guidelines is to assign default values for outputs in the beginning of the always block.



### Procedural blocks

### Initial block

☐ syntax

initial
begin
statements;
end

hint: begin and end are only needed when more than one statement inside initial block.

note: all variables assigned inside initial block are of type reg

- ☐ In initial block the statements are executed in sequence(procedurally).
- ☐ Initial block is used to describe the stimulus or test patterns and test benches for the design
- The initial block is executed once at time zero.
- ☐ Initial blocks are not synthesizable.
- # is used to suspend the execution of the initial block for a given delay.
- Timescale is a compiler directive specifies time unit and time precision for the modules

### Example

- ☐ The following circuit is a full-adder designed using the built-in gates.
- ☐ We will try to test this circuit using a testbench.



```
module Full_Adder_Structural_Verilog(
 input X1, X2, Cin,
 output S, Cout
  wire a1, a2, a3;
  xor u1(a1,X1,X2);
  and u2(a2,X1,X2);
  and u3(a3,a1,Cin);
  or u4(Cout,a2,a3);
  xor u5(S,a1,Cin);
endmodule
```

# Example

```
module full adder tb;
reg X1 tb, X2 tb, Cin tb;
wire S tb, Cout tb;
Full Adder Structural Verilog DUT(.X1(X1 tb), .X2(X2 tb),
.Cin(Cin tb),.S(S tb), .Cout(Cout tb));
initial
begin
 X1 tb=1'b0;
 X2 tb=1'b0;
 Cin tb=1'b0;
 #10
 $display("test case1");
 X1 tb=1'b1;
 X2 tb=1'b0;
 Cin tb=1'b1;
 #5
```

```
if (S_tb==0 && Cout_tb==1)
    $display("test case pass");
else
    $display("test case failed");
#10
 $display("test case2");
 X1 tb=1'b1;
 X2 tb=1'b1;
 Cin tb=1'b1;
 #5
 if(S tb==1 && Cout tb==1)
    $display("test case pass");
 else
     $display("test case failed");
 #5
 $stop;
end
endmodule
```

# ALU(Arithmetic Logic Unit)

### ALU (Arithmetic Logic Unit)

- □ An arithmetic-logic unit is the part of a central processing unit that carries out arithmetic and logic operations on the operands in computer instruction words.
- □ In some processors, the ALU is divided into two units: an arithmetic unit (AU) and a logic unit (LU).
- ☐ ALU do operations such as: add, sub, multiply,...(arithmetic operations) and OR, AND, XOR,....(logic operations).



### ALU (Arithmetic Logic Unit)

☐ Design an ALU that can execute the following operations and has a zero flag and a sign flag.

| Operation | binary |
|-----------|--------|
| A + B     | 000    |
| A SHL B   | 001    |
| В         | 010    |
| A - B     | 011    |
| A XOR B   | 100    |
| A SHR B   | 101    |
| A OR B    | 110    |
| A AND B   | 111    |



## Verilog Implementation

```
module ALU(
 input [31:0] A,B,
 input [2:0] sel,
 output [31:0] C,
 output ZF,SF
);
reg [32:0] R;
always@(*)
begin
 case(sel)
 3'b000 : R = A + B;
 3'b001 : R = A << B;
 3'b010 : R = B;
```

```
3'b011 : R = A - B;
 3'b100 : R = A \wedge B;
 3'b101 : R = A >> B;
 3'b110 : R = A \mid B;
 3'b111 : R = A \& B;
 default : R = 32'b0;
 endcase
end
assign C = R[31:0];
assign ZF = \sim(|R);
assign SF = R[32];
endmodule
```

### Citations

- Building a Half Adder | Coding projects for kids and teens (raspberrypi.org)
- full adder circuit diagram and code
- ALU circit block

### References

- > FPGA prototyping by Verilog examples Pong P. Chu.
- > VLSI Design Verilog Introduction VLSI tutorial.

# THANK YOU!